<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>

<div id="app">
    <!--计算属性-->
    <!--    模板内的表达式非常便利，但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。例如：-->
    {{ message.split('').reverse().join('') }}
    <!--    在这个地方，模板不再是简单的声明式逻辑。你必须看一段时间才能意识到，
    这里是想要显示变量 message 的翻转字符串。当你想要在模板中的多处包含此翻转字符串时，就会更加难以处理。-->
    <!--    所以，对于任何复杂逻辑，你都应当使用计算属性。-->

    <!--    基础例子-->
    <p> original message "{{ message }}"</p>
    <p>computed reversed message:"{{ reverseMessage }}"</p>

    <p>{{ now }}</p>
    <!--    这里我们声明了一个计算属性 reversedMessage。
    我们提供的函数将用作 property vm.reversedMessage 的 getter 函数：-->

    <!--    我们可以将同一函数定义为一个方法而不是一个计算属性。
    两种方式的最终结果确实是完全相同的。
    然而，不同的是计算属性是基于它们的响应式依赖进行缓存的。
    只在相关响应式依赖发生改变时它们才会重新求值。
    这就意味着只要 message 还没有发生改变，
    多次访问 reversedMessage 计算属性会立即返回之前的计算结果，而不必再次执行函数。-->

    <!--    计算属性vs侦听属性-->
    <!--    Vue 提供了一种更通用的方式来观察和响应 Vue 实例上的数据变动：侦听属性。
    当你有一些数据需要随着其它数据变动而变动时，
    你很容易滥用 watch——特别是如果你之前使用过 AngularJS。
    然而，通常更好的做法是使用计算属性而不是命令式的 watch 回调。细想一下这个例子：-->

    <p>{{ fullName }}</p>

</div>
<!--虽然计算属性在大多数情况下更合适，
但有时也需要一个自定义的侦听器。
这就是为什么 Vue 通过 watch 选项提供了一个更通用的方法，
来响应数据的变化。当需要在数据变化时执行异步或开销较大的操作时，这个方式是最有用的。-->
<div id="watch-example">
    <p>
        Ask a yes/no question:
        <input v-model="question">
    </p>
    <p>{{ answer }}</p>
</div>

<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
<script src="https://cdn.jsdelivr.net/npm/axios@0.12.0/dist/axios.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/lodash@4.13.1/lodash.min.js"></script>
<script>
    var vm = new Vue({
        el: "#app",
        data: {
            message: "i'm a girl",
            firstName: 'Foo',
            lastName: 'Bar',
            fullName: 'Foo Bar'
        },
        computed: {
            reverseMessage: function () {
                //this指向vm实例
                return this.message.split('').reverse().join('')
            },
            <!--    这也同样意味着下面的计算属性将不再更新，因为 Date.now() 不是响应式依赖：-->
            now: function () {
                return Date.now()
            },

//             相比之下，每当触发重新渲染时，调用方法将总会再次执行函数。
// 我们为什么需要缓存？假设我们有一个性能开销比较大的计算属性 A，
// 它需要遍历一个巨大的数组并做大量的计算。然后我们可能有其他的计算属性依赖于 A。
// 如果没有缓存，我们将不可避免的多次执行 A 的 getter！如果你不希望有缓存，请用方法来替代。


            // fullName: function () {
            //     return this.firstName + ' ' + this.lastName
            // },

            //    计算属性默认只有getter，不过在需要时你也可以提供一个setter
            fullName: {
                get: function () {
                    return this.firstName + this.lastName
                },
                set: function (newValue) {
                    var names = newValue.split(' ');
                    this.firstName = names[0];
                    this.lastName = names[names.length - 1];
                }
                //    现在再运行 vm.fullName = 'John Doe' 时，
                //    setter 会被调用，vm.firstName 和 vm.lastName 也会相应地被更新。
            }
        },
        watch: {
            firstName: function (val) {
                this.fullName = val + ' ' + this.lastName
            },
            lastName: function (val) {
                this.fullName = this.firstName + ' ' + val
            }
            //    上面代码是命令式且重复的。将它与计算属性的版本进行比较：
        }
    })
    //在这个示例中，使用 watch 选项允许我们执行异步操作 (访问一个 API)，
    // 限制我们执行该操作的频率，并在我们得到最终结果前，
    // 设置中间状态。这些都是计算属性无法做到的。
    //除了 watch 选项之外，您还可以使用命令式的 vm.$watch API。
    var watchExampleVM = new Vue({
        el: "#watch-example",
        data: {
            question: '',
            answer: 'i cannot give you a answer until you ask a question!'
        },
        watch: {
            //如果'question'发生改变,这个函数就会运行
            question: function (newQuestion, oldQuestion) {
                this.answer = 'Waiting for you to stop typing...'
                this.debouncedGetAnswer()
            }
        },
        created:
            function () {
                // `_.debounce` 是一个通过 Lodash 限制操作频率的函数。
                // 在这个例子中，我们希望限制访问 yesno.wtf/api 的频率
                // AJAX 请求直到用户输入完毕才会发出。想要了解更多关于
                // `_.debounce` 函数 (及其近亲 `_.throttle`) 的知识，
                // 请参考：https://lodash.com/docs#debounce
                this.debouncedGetAnswer = _.debounce(this.getAnswer, 500)
            }
        ,
        methods: {
            getAnswer: function () {
                if (this.question.indexOf('?') === -1) {
                    this.answer = 'Question usually contain a question mark.;-)'
                    return
                }
                this.answer = 'Thinking...'
                var vm = this
                axios.get('https://yesno.wtf/api').then(function (response) {
                    vm.answer = _.capitalize(response.data.answer)
                }).catch(function (error) {
                    vm.answer = 'Error!Cound not reach the API.' + error
                })

            }

        }
    })
</script>
</body>
</html>